### Rechnerstrukturen im SS2007

#### Hardwareentwurf

### Dr. Rainer Buchty

buchty@ira.uka.de

Universität Karlsruhe (TH) – Forschungsuniversität Institut für Technische Informatik (ITEC) Lehrstuhl für Rechnerarchitektur und Parallelverarbeitung

31.05.2007

# Die HW-Beschreibungssprache VHDL

- VHDL = VHSIC HDL
- VHSIC = Very High Speed Integrated Circuits
- VHDL hervorgegangen aus dem VHSIC-Programm der USA
- Standardisierte Hardware-Beschreibungssprache
- Seit 1987 IEEE-Standard, mittlerweile überarbeitet
- Kann verschiedene Beschreibungen des gesamten Entwurfsablauf darstellen
  - Algorithmische Spezifikation (behavioral)
  - Realisierungsnahe Strukturen (RTL)
  - Simulation
- Vorsicht: Nicht alle Sprachkonstrukte in jedem Entwurfsschritt nutzbar!

# VHDL (forts.)

### Die Entwurfssprache VHDL

- Ursprünglich als Modellierungssprache nur für die Simulation konzipiert
- Zunehmender Einsatz als Sprache für Synthese und Verifikation
- heute: VHDL, Verilog, SystemC
- Einsatz zum ASIC- und FPGA-Entwurf, mit Erweiterungen auch Analog-Entwurf möglich
- VHDL umfaßt alle Elemente einer klassischen Programmiersprache (ADA)
- Erweitert um Konstrukte für Schaltungsentwurf

# VHDL (forts.)

#### VHDL-Fallstricke

- Mächtigkeit der Sprache problematisch:
- Nicht alle Sprachkonstrukte k\u00f6nnen in Hardware umgesetzt werden
- Trennung zwischen simulierbar und synthetisierbar
- Abstraktion erzeugt eine im Vergleich zu niederen Modellierungssprachen (ABEL, PALASM) – gewissen Mehraufwand in der Beschreibung
- Methoden der Abstraktion für Ein- und Umsteiger ggf. gewöhnungsbedürftig (z.B. Registerbeschreibung)

# Chipentwurf mit VHDL

### Grundlage: Spezifikation der Schaltung

- Schnittstellen (Zahl und Art der Ein-/Ausgänge)
- Gewünschtes Verhalten
- Eventuelle weitere Vorgaben bezüglich Geschwindigkeit, Kosten, Fläche, Leistungsverbrauch etc.

- Nur Schnittstellen und gewünschtes Verhalten werden typischerweise in VHDL formuliert
- Geschwindigkeitsvorgaben nur simulierbar (Einhalten von Zeitfenstern)
- Weitere Parameter Domäne des Synthesewerkzeugs

### Entwurfsschritte

### Verhaltensverfeinerung

- Detailliertes Ausarbeiten der gewünschten Funktionalität
- Ersetzen von Black-Boxes

### Strukturverfeinerung

 Realisierung einer spezifizierten Funktion durch Verschaltung von Komponenten mit einfacherer Funktionalität

### Datenverfeinerung

- Realisierung abstrakter Datentypen durch einfachere Datentypen
- Synthese: Binärer Datentyp bzw. erweiterter binärer Datentyp (incl. Tri-state-Zustand, std\_logic)

### → Schaltungsbeschreibung



# Erstellen einer Beschreibung

### Schnittstellendefinition (entity)

- Entity beschreibt Ein-/Ausgabeschnittstelle
- Pro Modul nur eine Entity erlaubt

### Verhaltensbeschreibung (architecture)

- Mehrere Architectures pro Modul möglich, z.B. zur Unterscheidung von Verhaltens- oder Synthesebeschreibung
- Keine begriffliche Verwandtschaft zu (Mikro)Architektur bei Prozessoren

### **Konfiguration und Zuordnung (configuration)**

- Festlegung verwendeter Architecture
- Konfiguration, z.B. von Signalbreiten



# Signalmodi

### Signalmodus bestimmt Datenflußrichtung

- in kann nur gelesen werden
- out kann nur geschrieben werden
- inout bezeichnet bidirektionales Signal
- buffer ist ebenfalls bidirektional mit eingeschränkter
   Zuweisungsmöglichkeit (VHDL-FAQ: "The use of buffer ports is discouraged")
- linkage dient zur Verbindung mit externen, nicht-VHDL Modulen; Semantik werkzeug- bzw. herstellerspezifisch
- Typischerweise nur in, out, inout in Verwendung
- Modi werden nur in Entity deklariert, nicht bei internen (in der Architecture spezifizierten) Signalen

# Datentypen

### VHDL-Datentypen

- boolean: True, False
- bit: 0, 1
- std\_logic
  - Erweiterung von bit mit zusätzlich
  - Z: tristate / hochohmig; Signal mit diesem Wert kann von anderen Signalen mit 0 oder 1 überschrieben werden
  - X: unbekannt; Datenwert ist durch unvollständige oder fehlerhafte Berechung / Zusammenschaltung entstanden
  - U: undefiniert, d.h. uninitialisiert oder unbekannt
- Skalare Datentypen, Arrays (\_vector), Integer, Characters
- Fließkomma, Datei, Records, eigene Typdefinitionen
- Synthetisierbarkeit beachten!

# Zuweisungsoperatoren

- Zuweisung von skalaren Werten
  - a<='1';
- Zuweisung von Vektoren
  - a<="1010":
  - a<=(others=>'0');
- Zuweisung von Signalen
  - a<=b;
- Zuweisung von Termen
  - a<=not(b);</p>

#### Zu beachten:

- Zuweisungsoperator unterscheidet zwischen Signalen <= und Variablen :=</li>
- VHDL führt strikte Typprüfung durch (auch für Operatoren)
- Ggf. Konvertierung (typecasting) notwendig

## Operatoren

#### **VHDL-Operatoren**

- Logik: AND, OR, NAND, NOR, XOR, XNOR
- Vergleich: =, / =, <, <=, >, >=
- Schieben: SLL, SRL, SLA, SRA, ROL, ROR
- Arithmetik: +, -, \*, /, MOD
- Diverse: \*\*, ABS, NOT
- Wichtig: Operatoren innerhalb einer Gruppe haben gleiche Präzedenz
- AND/OR haben gleiche Priorität, d.h. gemischte Ausdrücke passend klammern, um Mehrdeutigkeiten zu vermeiden
- Ordnung der Gruppen nach aufsteigender Präzedenz
- Datentypen und Operatoren werden durch Bibliothek definiert; erst durch Benutzung von entsprechenden Bibliotheken werden Datentypen und Operatoren erst nutzbar

### Auch hier gilt: nicht zwingend in Hardware abbildbar



# Datenobjekte

 Datenobjekte haben Typ und – je nach Deklarationsort – Modus

### Signale (signals)

- Häufigstes Datenobjekt
- Datentransport und Datenspeicherung innerhalb von Architectures und über deren Grenzen (Schnittstellen) hinaus.
- Wertezuweisung nebenläufig, wenn nicht explizit serialisiert (Prozesse)

### Konstanten (constants)

Statische Werte

# Datenobjekte

#### Variablen (variables)

- Definition innerhalb des process-Headers
- Auf einzelnen Process beschränkt, kein Transport über process-Grenzen hinweg
- Wertezuweisung sequentiell und umittelbar in der Reihenfolge der Abarbeitung
- Unterschied zu Signalen!

# Entwurfsbeispiel: NAND-Gatter

$$NAND: f(x,y) = \begin{cases} 0 & wenn \ x = 1 \land y = 1 \\ 1 & sonst \end{cases}$$

- Datentyp std\_logic soll verwendet werden
- Einladen benötigter Bibliotheken

### NAND-Gatter: Einladen benötigter Bibliotheken

```
library IEEE;
use IEEE.std_logic_1164.all;
```

$$NAND: f(x,y) = \begin{cases} 0 & wenn \ x = 1 \land y = 1 \\ 1 & sonst \end{cases}$$

- Zwei Eingänge x/y, ein Ausgang f(x,y)
- Interfacebeschreibung ergibt Entity

### NAND-Gatter: Beschreibung der Schnittstellen

```
entity NAND is
  port(
    -- x,y sind Eingänge vom Typ std_logic
    x,y: in std_logic;
    -- fxy ist Ausgang vom Typ std_logic
    fxy: out std_logic
);
end entity;
```

$$NAND: f(x,y) = \begin{cases} 0 & wenn \ x = 1 \land y = 1 \\ 1 & sonst \end{cases}$$

Funktionsbeschreibung ergibt Architecture

### NAND-Gatter: Beschreibung des Verhaltens

```
architecture arch_NAND of NAND is
begin
    -- funktionale Beschreibung
    fxy<='0' when x='1' and y='1' else '1';
end architecture;</pre>
```

$$NAND: f(x,y) = \begin{cases} 0 & wenn \ x = 1 \land y = 1 \\ 1 & sonst \end{cases}$$

• Funktionsbeschreibung ergibt Architecture

### NAND-Gatter: Beschreibung des Verhaltens

```
architecture arch_NAND of NAND is
-- Hilfssignal
signal f_and: std_logic;
begin
    -- algorithmische Beschreibung
    f_and<=x and y;
    fxy<=not(f_and);
end architeture;</pre>
```

$$NAND: f(x,y) = \begin{cases} 0 & wenn \ x = 1 \land y = 1 \\ 1 & sonst \end{cases}$$

- Weiterverwendung des NAND-Gatters zur Instantiierung in anderen Architectures (port map-Statement)
- Komponentenbeschreibung
- Ggf. Zusammenfassung mehrer Komponenten in eigenem package zum Aufbau einer Bibliothek ( → use-Statement)

### NAND-Gatter: Komponentenbeschreibung

```
component NAND is
  port(
    x,y: in std_logic;
    fxy: out std_logic
);
end component;
```

# Prozesse und Sequentialität

- Zuweisungen in VHDL grundsätzlich nebenläufig
- Modellierung von Schaltwerken explizit in Prozessen (process)
- Beschreibung des Verhaltens über einen sequentiellen Algorithmus
- Simulierte Zeit für die gesamte VHDL-Beschreibung schreitet während Ausführung nicht fort

#### **Aufbau eines Prozesses**

- Sensitivity List; Signale, bei deren Änderung der Beschreibungsblock ausgeführt wird
- Rückhalten der Zuweisungen bis Blockende
- Variablendeklaration (:=), Zuweisung sofort
- Verhaltensbeschreibung

# Prozeß-Sprachkonstrukte

### Warten auf Ereignis

- wait for
- wait until
- Problematisch bei Synthese!

### **Bedingte Zuweisung**

- if/then/else (entspricht when/then-Konstrukt bei nebenläufiger Zuweisung)
- case/when-Zuweisung

#### **Schleifen**

- for/loop
- while/loop
- Auch Zuweisungen innerhalb einer Schleife erfolgen erst zum Ende des Blocks
- Ebenfalls problematisch bei Synthese

# Modellierung von Speicherelementen

 VHDL verfügt über kein explizites Sprachelement zur Erzeugung von Speicherelementen

### Modellierung von Speicherelementen

- Erzeugung über if-Konstrukt in Prozessen
- Asynchrones Speicherelement (Latch, pegelgesteuert)
- Synchrones Speicherelement (Flip-Flop, taktflankengesteuert)
- Spezifikation über Signalattribut ('event) und Signalpegel
- Pro process und Taktsignal kann nur eine Taktflanke herangezogen werden
- Speicherelementtyp (D, T, RS, JK) mittels Ausformulierung des gewünschten Verhaltens

# Speicherlemenente (forts.)

- Beispiel: D-Latch
- Einspeichern von Daten während low-Pegel des Kontrollsignals möglich
- Verriegelt w\u00e4hrend high-Pegel

### **Modellierung eines D-Latchs**

```
D_LATCH:
process(write,data)
begin
  if write='0' then
    dlatch<=data;
  end if;
end process;</pre>
```

# Speicherlemenente (forts.)

- Beispiel: D-Flipflop
- Übernahme von Daten zur steigenden Taktflanke eines Taktsignals

### **Modellierung eines D-Flipflops**

```
D.FF:
process(clk,data)
begin
  if clk'event and clk='1' then
    dff<=data;
  end if;
end process;</pre>
```

# Speicherlemenente (forts.)

- Beispiel: D-Flipflop mit asynchronem Rücksetzeingang
- Übernahme von Daten zur steigenden Taktflanke eines Taktsignals
- Löschen des Inhaltes asynchron, d.h. unabhängig vom Taktsigal, bei low-Pegel des Rücksetzsignals

### **Modellierung eines D-Flipflops**

```
D_FFR:
process(rst,clk,data)
begin
  if rst='0' then
    dffr<=(others=>'0');
  elsif clk'event and clk='1' then
    dffr<=data;
  end if;
end process;</pre>
```

# Entwurfsbeispiel: Register-File

- 16 Register zu je 8-Bit
- Schreib- und lesbar
- Steuersignale: Takt, Rücksetzeingang, Chip-Select
- Zugriff via Adreß/Datenbus

### Register-File: Interface

```
entity register is
  port(
    clk, rst, cs, rw: in std_logic;
    addr: in std_logic_vector(3 downto 0);
    data: inout std_logic_vector(7 downto 0)
  );
end entity;
```

- 16 8-Bit Register → zweidimensionales Array
- In VHDL nicht direkt abbildbar, d.h. zweistufiges Verfahren
  - Definition eines Subtypen für 8-Bit Arrays
  - Signaldeklaration
- Ausgabe nicht direkt auf Datenbus sondern in internes Signal

### Register-File: Deklarationen

```
architecture arch_reg of register is
subtype sdlv8 is std_logic_vector(7 downto 0);
signal register: sdlv8_vector(15 downto 0);
signal data_out: std_logic_vector(7 downto 0);
begin
...
end architecture;
```

- Prozeßdeklaration:
  - Steuersignale sind Takt, Reset, Chip-Select und R/W
  - Auswahl mittels Adreßleitungen

### Register-File: Deklarationen und Reset)

```
process(clk,rst,cs,rw,addr)
begin
if rst='0' then
  register(0) <= (others=>'0');
  register(1) <= (others=>'0');
  ...
  -- alternativ: generate-Statement
```

- Zugriff synchron zu Takt
- Zugriffsart bestimmt durch R/W
- Gültigkeit des Zugriffs durch Chip-Select
- Schreiben synchron, Lesen asynchron

### Register-File: Zugriffe

```
elsif clk'event and clk='1' then
    if rw='0' and cs='0' then
        -- Datum in Register schreiben
    end if;
    end if;
    end process;
    -- Datum aus Register lesen
end architecture;
```

- Register lesen: nebenläufig, nicht taktflankengesteuert
- Ausgabe auf Datenbus gesteuert durch R/W und Chip-Select
- Nebenläufige Zuweisung außerhalb des Prozesses

### Register-File: Asynchroner Lesezugriff

```
data<=data_out when rw='1' and cs='0'
else (others=>'Z');
data_out<=register(0) when addr="0000"
        else register(1) when addr="0001"
        ...</pre>
```

- Register schreiben: synchron zu Taktflanke
- Plazierung innerhalb des Prozesses

### Register-File: Synchroner Schreibzugriff

```
case addr is
  when "0000" => register(0) <= data;
  when "0001" => register(1) <= data;
  ...
  when others => null;
end case;
```

### Weiterführende Literatur

#### Literatur

 VHDL-Kurzanleitung von Georg Acher und Markus Leberecht

http://www.lrr.in.tum.de/~acher/tgi/uebung/VHDL-Buch.pdf

 VHDL-Cookbook von Peter J. Ashenden mit detaillierter Einführung und sehr ausführlichem Entwurfsbeispiel (DP32-Prozessor)

http://tech-www.informatik.uni-hamburg.de/vhdl/doc/cookbook/ VHDL-Cookbook.pdf

#### Für Interessierte

The Hamburg VHDL Archive

http://tams-www.informatik.uni-hamburg.de/vhdl/

Freie IP-Cores in VHDL und Verilog

http://www.opencores.org



# Aufgabe

Eine Zählerschaltung soll entwickelt werden. Diese Schaltung soll folgendes Verhalten aufweisen:

- Ein low-aktives Rücksetzsignal löscht den Zähler.
- Über ein Richtungssignal wird bestimmt, ob der Zähler mit der steigenden Flanke eines Taktsignals aufwärts (=0) oder abwärts (=1) zählt.
- Es wird nur gezählt, wenn der Zähler mit einem high-aktiven Auswahl-Signal freigeschaltet ist.
- Der Zähler soll 64 Zählschritte ausführen können.
- Ein low-aktives Freigabesignal entscheidet, ob der Zählerausgang auf einen gemeinsamen Bus gelegt werden soll; bei nicht erfolgter Freigabe werden die Ausgabeleitungen in den tri-state-Zustand geschaltet.

# Aufgabe (forts.) – entity

 Erstellen Sie die zugehörige Schnittstellenbeschreibung in VHDL.

# Aufgabe (forts.) – entity

 Erstellen Sie die zugehörige Schnittstellenbeschreibung in VHDL.

Schnittstelle enthält alle nach außen sichtbaren Signale mit Datentyp und Richtung, also:

```
entity counter is
  port(
    rst: in std_logic; -- Rücksetzsignal
    clk: in std_logic; -- Taktsignal
    dir: in std_logic; -- Richtungsumschaltung
    sel: in std_logic; -- Auswahlsignal
    ena: in std_logic; -- Freigabesignal

    c_out: out std_logic_vector(5 downto 0) -- Zählerausgabe
    );
end entity;
```

Es ist alternativ auch möglich, Signale gleichen Typs und Richtung zu gruppieren, z.B. rst, clk, dir, sel, ena: in std\_logic;

# Aufgabe (forts.) – architecture

 Formulieren Sie die entsprechende Verhaltensbeschreibung in VHDL.

# Aufgabe (forts.) – architecture

 Formulieren Sie die entsprechende Verhaltensbeschreibung in VHDL.

Problemfall: c\_out ist als reines Ausgabesignal deklariert und kann nicht als eigentlicher Zähler verwendet werden. In der Verhaltensbeschreibung muß zusätzlich der eigentliche Zähler als Signal deklariert werden. Die Ausgabe des Zählers erfolgt außerhalb des Prozesses.

# Aufgabe (forts.) – architecture

 Formulieren Sie die entsprechende Verhaltensbeschreibung in VHDL.

Problemfall: c\_out ist als reines Ausgabesignal deklariert und kann nicht als eigentlicher Zähler verwendet werden. In der Verhaltensbeschreibung muß zusätzlich der eigentliche Zähler als Signal deklariert werden. Die Ausgabe des Zählers erfolgt außerhalb des Prozesses.

```
architecture arch_counter of counter is
  signal count: unsigned(5 downto 0); -- 64 Zustände
begin
  -- Ausgabe
  c_out<=count when ena='0' else (others=>'Z');
end architecture;
```

## Aufgabe (forts.) – architecture/process

# Prozeß zerfällt in asynchrones Rücksetzen und eigentlichen Zählvorgang.

```
-- Zählfunktion
p_counter:
process(rst,clk,dir,sel)
                               elsif clk'event and clk='1' then
begin
                               -- Zähler selektiert?
  -- asynchrones Rücksetzen
                                 if sel='1' then
  if rst='0' then
    count <= (others=>'0'); -- Zählrichtung
                                    if dir='0' then
                                      count <= count +1;
                                    else
                                      count <= count -1;
                                    end if;
                                 end if:
                               end if;
                             end process;
```

# Aufgabe (forts.)

Der Zähler soll um eine Über-/Unterlaufsfunktion ergänzt werden, d.h. beim Erreichen des Wertes 0 wird für die Dauer eines Taktes ein Anzeigesignal ausgegeben; hierbei soll das Rücksetzsignal nicht fälschlicherweise das Überlaufssignal auslösen. In der Entity sei hierzu das zusätzliche signal ovl vom Typ std\_logic mit Modus out deklariert.

- Erweitern Sie die Verhaltensbeschreibung aus Teilaufgabe b) um eine Lösung, bei der das Überlaufssignal rein innerhalb des Prozesses erzeugt wird. Was ist bei dieser Lösung zu beachten und warum? Wozu würde ovl in dieser Implementation synthetisiert?
- Erweitern Sie die Verhaltensbeschreibung aus Teilaufgabe
   b) um eine Lösung, bei der das Überlaufssignal außerhalb des Prozesses erzeugt wird.
- → Lösungsblatt

